<!DOCTYPE html>



  


<html class="theme-next gemini use-motion" lang="zh-Hans">
<head><meta name="generator" content="Hexo 3.8.0">
  <meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
<meta name="theme-color" content="#222">



  
  
    
    
  <script src="/lib/pace/pace.min.js?v=1.0.2"></script>
  <link href="/lib/pace/pace-theme-minimal.min.css?v=1.0.2" rel="stylesheet">







<meta http-equiv="Cache-Control" content="no-transform">
<meta http-equiv="Cache-Control" content="no-siteapp">






















<link href="/lib/font-awesome/css/font-awesome.min.css?v=4.6.2" rel="stylesheet" type="text/css">

<link href="/css/main.css?v=5.1.4" rel="stylesheet" type="text/css">


  <link rel="apple-touch-icon" sizes="180x180" href="/images/favicon-32x32-st.png?v=5.1.4">


  <link rel="icon" type="image/png" sizes="32x32" href="/images/favicon-32x32-st.png?v=5.1.4">


  <link rel="icon" type="image/png" sizes="16x16" href="/images/favicon-32x32-st.png?v=5.1.4">


  <link rel="mask-icon" href="/images/logo.svg?v=5.1.4" color="#222">





  <meta name="keywords" content="读书笔记,《JavaScript 高级程序设计》,">





  <link rel="alternate" href="/atom.xml" title="卷柏的花期" type="application/atom+xml">






<meta name="description" content="本篇包含了每一门（不仅限于JavaScript）计算机语言的最基本组成部分，比如识符与变量、关键字与保留字、运算符与表达式、数据类型以及程序语句等。还重点讲述了数据类型中 undefined与 null之间的区别，浮点型精度丢失的原理和解决方案，不同类型之间运算的隐式转换问题…..">
<meta name="keywords" content="读书笔记,《JavaScript 高级程序设计》">
<meta property="og:type" content="article">
<meta property="og:title" content="《JavaScript 高级程序设计》第三章：基本概念">
<meta property="og:url" content="http://yoursite.com/2018/06/01/professional-javascript-chapter-3/index.html">
<meta property="og:site_name" content="卷柏的花期">
<meta property="og:description" content="本篇包含了每一门（不仅限于JavaScript）计算机语言的最基本组成部分，比如识符与变量、关键字与保留字、运算符与表达式、数据类型以及程序语句等。还重点讲述了数据类型中 undefined与 null之间的区别，浮点型精度丢失的原理和解决方案，不同类型之间运算的隐式转换问题…..">
<meta property="og:locale" content="zh-Hans">
<meta property="og:image" content="http://yoursite.com/2018/06/01/professional-javascript-chapter-3/javascript.jpg">
<meta property="og:image" content="http://yoursite.com/2018/06/01/professional-javascript-chapter-3/ecmascript-type.png">
<meta property="og:image" content="http://yoursite.com/2018/06/01/professional-javascript-chapter-3/ieee754.png">
<meta property="og:image" content="http://yoursite.com/2018/06/01/professional-javascript-chapter-3/8-ym.png">
<meta property="og:updated_time" content="2018-12-31T09:16:03.495Z">
<meta name="twitter:card" content="summary">
<meta name="twitter:title" content="《JavaScript 高级程序设计》第三章：基本概念">
<meta name="twitter:description" content="本篇包含了每一门（不仅限于JavaScript）计算机语言的最基本组成部分，比如识符与变量、关键字与保留字、运算符与表达式、数据类型以及程序语句等。还重点讲述了数据类型中 undefined与 null之间的区别，浮点型精度丢失的原理和解决方案，不同类型之间运算的隐式转换问题…..">
<meta name="twitter:image" content="http://yoursite.com/2018/06/01/professional-javascript-chapter-3/javascript.jpg">



<script type="text/javascript" id="hexo.configurations">
  var NexT = window.NexT || {};
  var CONFIG = {
    root: '/',
    scheme: 'Gemini',
    version: '5.1.4',
    sidebar: {"position":"left","display":"post","offset":12,"b2t":false,"scrollpercent":false,"onmobile":false},
    fancybox: false,
    tabs: true,
    motion: {"enable":true,"async":false,"transition":{"post_block":"fadeIn","post_header":"slideDownIn","post_body":"slideDownIn","coll_header":"slideLeftIn","sidebar":"slideUpIn"}},
    duoshuo: {
      userId: '0',
      author: '博主'
    },
    algolia: {
      applicationID: '',
      apiKey: '',
      indexName: '',
      hits: {"per_page":10},
      labels: {"input_placeholder":"Search for Posts","hits_empty":"We didn't find any results for the search: ${query}","hits_stats":"${hits} results found in ${time} ms"}
    }
  };
</script>



  <link rel="canonical" href="http://yoursite.com/2018/06/01/professional-javascript-chapter-3/">





  <title>《JavaScript 高级程序设计》第三章：基本概念 | 卷柏的花期</title>
  








</head>

<body itemscope="" itemtype="http://schema.org/WebPage" lang="zh-Hans">

  
  
    
  

  <div class="container sidebar-position-left page-post-detail">
    <div class="headband">
    <a href="https://gitee.com/Selaginella/" target="_blank" style="position:absolute;right:0;top:0;border:none"><img src="https://gitee.com/Selaginella/Selaginella/widgets/widget_1.svg" alt="Fork me on Gitee"></a>
    </div>

    <header id="header" class="header" itemscope="" itemtype="http://schema.org/WPHeader">
      <div class="header-inner"><div class="site-brand-wrapper">
  <div class="site-meta ">
    

    <div class="custom-logo-site-title">
      <a href="/" class="brand" rel="start">
        <span class="logo-line-before"><i></i></span>
        <span class="site-title">卷柏的花期</span>
        <span class="logo-line-after"><i></i></span>
      </a>
    </div>
      
        <p class="site-subtitle">随风而动，九死还生</p>
      
  </div>

  <div class="site-nav-toggle">
    <button>
      <span class="btn-bar"></span>
      <span class="btn-bar"></span>
      <span class="btn-bar"></span>
    </button>
  </div>
</div>

<nav class="site-nav">
  

  
    <ul id="menu" class="menu">
      
        
        <li class="menu-item menu-item-home">
          <a href="/" rel="section">
            
              <i class="menu-item-icon fa fa-fw fa-home"></i> <br>
            
            首页
          </a>
        </li>
      
        
        <li class="menu-item menu-item-archives">
          <a href="/archives/" rel="section">
            
              <i class="menu-item-icon fa fa-fw fa-archive"></i> <br>
            
            归档
          </a>
        </li>
      
        
        <li class="menu-item menu-item-about">
          <a href="/about/" rel="section">
            
              <i class="menu-item-icon fa fa-fw fa-user"></i> <br>
            
            关于
          </a>
        </li>
      
        
        <li class="menu-item menu-item-source">
          <a href="/source/" rel="section">
            
              <i class="menu-item-icon fa fa-fw fa-flask"></i> <br>
            
            资源
          </a>
        </li>
      

      
        <li class="menu-item menu-item-search">
          
            <a href="javascript:;" class="popup-trigger">
          
            
              <i class="menu-item-icon fa fa-search fa-fw"></i> <br>
            
            搜索
          </a>
        </li>
      
    </ul>
  

  
    <div class="site-search">
      
  <div class="popup search-popup local-search-popup">
  <div class="local-search-header clearfix">
    <span class="search-icon">
      <i class="fa fa-search"></i>
    </span>
    <span class="popup-btn-close">
      <i class="fa fa-times-circle"></i>
    </span>
    <div class="local-search-input-wrapper">
      <input autocomplete="off" placeholder="搜索..." spellcheck="false" type="text" id="local-search-input">
    </div>
  </div>
  <div id="local-search-result"></div>
</div>



    </div>
  
</nav>



 </div>
    </header>

    <main id="main" class="main">
      <div class="main-inner">
        <div class="content-wrap">
          <div id="content" class="content">
            

  <div id="posts" class="posts-expand">
    

  

  
  
  

  <article class="post post-type-normal" itemscope="" itemtype="http://schema.org/Article">
  
  
  
  <div class="post-block">
    <link itemprop="mainEntityOfPage" href="http://yoursite.com/2018/06/01/professional-javascript-chapter-3/">

    <span hidden itemprop="author" itemscope="" itemtype="http://schema.org/Person">
      <meta itemprop="name" content="Selaginella">
      <meta itemprop="description" content="">
      <meta itemprop="image" content="/uploads/avatar.jpg">
    </span>

    <span hidden itemprop="publisher" itemscope="" itemtype="http://schema.org/Organization">
      <meta itemprop="name" content="卷柏的花期">
    </span>

    
      <header class="post-header">

        
        
          <h1 class="post-title" itemprop="name headline">《JavaScript 高级程序设计》第三章：基本概念</h1>
        

        <div class="post-meta">
          <span class="post-time">
            
              <span class="post-meta-item-icon">
                <i class="fa fa-calendar-o"></i>
              </span>
              
                <span class="post-meta-item-text">发表于</span>
              
              <time title="创建于" itemprop="dateCreated datePublished" datetime="2018-06-01T20:43:00+08:00">
                2018-06-01
              </time>
            

            

            
          </span>

          
            <span class="post-category">
            
              <span class="post-meta-divider">|</span>
            
              <span class="post-meta-item-icon">
                <i class="fa fa-folder-o"></i>
              </span>
              
                <span class="post-meta-item-text">分类于</span>
              
              
                <span itemprop="about" itemscope="" itemtype="http://schema.org/Thing">
                  <a href="/categories/JavaScript/" itemprop="url" rel="index">
                    <span itemprop="name">JavaScript</span>
                  </a>
                </span>

                
                
              
            </span>
          

          
            
          

          
          

          

          

          

        </div>
      </header>
    

    
    
    
    <div class="post-body" itemprop="articleBody">

      
      

      
        <blockquote class="blockquote-center"><img src="/2018/06/01/professional-javascript-chapter-3/javascript.jpg" alt="&#39;JavaScript 高级程序设计&#39;">本篇包含了每一门（不仅限于JavaScript）计算机语言的最基本组成部分，比如识符与变量、关键字与保留字、运算符与表达式、数据类型以及程序语句等。还重点讲述了数据类型中 <code>undefined</code>与 <code>null</code>之间的区别，浮点型精度丢失的原理和解决方案，不同类型之间运算的隐式转换问题…..</blockquote>

<a id="more"></a>
<hr>
<h2 id="语法"><a href="#语法" class="headerlink" title="语法"></a>语法</h2><p>“语法”指的是一门语言的书写风格，JavaScript 的语法风格很类似于 C 以及 Java。<br>语法又是一种概念上的统称，例如如何声明变量、定义函数、书写流程控制语句或者是进行条件判断等，这些都是属于语法风格的范畴。</p>
<h2 id="标识符"><a href="#标识符" class="headerlink" title="标识符"></a>标识符</h2><p>所谓的“标识符”就是对变量、标签、函数、方法、属性以及参数名称的统称。<br>标识符可以分为“用户自定义标识符”与“系统标识符”，其中对与系统标识符，我们可以根据其使用状况，又划分为“关键字”或者是“保留字”。<br>用户自定义标识符，其命名应遵循以下规则：</p>
<ul>
<li>开头只能以字母，下划线(_)、美元符号($)、π等字符开头。</li>
<li>除了开头的字符外，标识符的其它字符可以是字母、数字、下划线、美元符号以及其它ASCII码字符或者是扩展ASCII码字符（当然Unicode中的字符也是可以的，但并不建议这么做）</li>
<li>在命名的格式上，我们强烈建议驼峰命名法。</li>
</ul>
<h2 id="严格模式"><a href="#严格模式" class="headerlink" title="严格模式"></a>严格模式</h2><p>通过声明 <code>use strict</code> 可以让支持严格模式的 JavaScript引擎切换到另一种解析与执行模式，在严格模式下 ES3中的一些不确定行为将会按照ES5(ES3.1)的规定去处理。<br><code>use strict</code> 如果声明在代码的第一行，则整个JS程序都会按照严格模式去执行，如果你只想在某个局部区域使用该模式，请将 <code>use strict</code> 单独加在某个局部作用域的开头。</p>
<h2 id="关键字"><a href="#关键字" class="headerlink" title="关键字"></a>关键字</h2><p>用于特定用途的标识符，这些标识符可用于表示控制流程的开始或结束，或者用于执行特定的操作，例如声明变量、定义函数等。</p>
<h2 id="保留字"><a href="#保留字" class="headerlink" title="保留字"></a>保留字</h2><p>保留字是语言层面保留的标识符，它与关键字的主要区别就在于作为保留字的标识符暂时还未指定用途，但它们未来很可能会用作于关键字。</p>
<h2 id="变量"><a href="#变量" class="headerlink" title="变量"></a>变量</h2><p>ECMAScript中的变量是弱类型的，相当于保存值的占位符。<br>声明但未初始化的变量，其默认值为 <code>undefined</code> 。<br>ECMAScript支持同时声明并初始化多个变量，每个变量用逗号隔开</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> obj = &#123;&#125;,</span><br><span class="line">    str = <span class="string">'string'</span>,</span><br><span class="line">    num = <span class="number">0</span>,</span><br><span class="line">    nul = <span class="literal">null</span>,</span><br><span class="line">    und = <span class="literal">undefined</span>,</span><br><span class="line">    boolean = <span class="literal">true</span>;</span><br></pre></td></tr></table></figure>
<blockquote>
<p>在ECMAScript中不通过 <code>var</code> 关键字声明的变量，默认会附加到全局 <code>window</code> 对象上。</p>
</blockquote>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line">a = <span class="number">1</span>;</span><br><span class="line"><span class="built_in">window</span>.a <span class="comment">// 1</span></span><br></pre></td></tr></table></figure>
<blockquote>
<p>该种方式在 ECMAScript 5 的严格模式(strict)下是不被允许的。</p>
</blockquote>
<h2 id="数据类型"><a href="#数据类型" class="headerlink" title="数据类型"></a>数据类型</h2><p>下面是ECMAScript5 的数据类型：</p>
<p><img src="/2018/06/01/professional-javascript-chapter-3/ecmascript-type.png" alt="&#39;数据类型&#39;"></p>
<p>对于数据类型的检测，JavaScript中也提供了一个内置的操作符 — <code>typeof</code>，通过 <code>typeof</code> 我们可以准确的获得 “string”、”number”、”boolean”、”undefined” 等基本数据类型，但是需要注意的是使用 <code>typeof</code> 对 <code>null</code> 或者其它对象型或者是数组检测的时候，统一都返回 <code>object</code> 类型。</p>
<h3 id="Undefined"><a href="#Undefined" class="headerlink" title="Undefined"></a>Undefined</h3><p><code>undefined</code> 类型只有一个值，那就是：”undefined”。<br>一般而言我们声明未初始化的变量其默认值就是 <code>undefined</code>，而ECMAScript引入 <code>undefined</code> 的作用就是为了更好的与 <code>null</code> 进行区分。<br>这里有一个技巧，使用 <code>typeof</code> 操作符来判断未声明的变量，并不会如我们想的那样会报错，而是返回 <code>undefined</code> 值。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">typeof</span> und <span class="comment">// undefined</span></span><br></pre></td></tr></table></figure>
<h3 id="Null"><a href="#Null" class="headerlink" title="Null"></a>Null</h3><p><code>null</code> 与 <code>undefined</code> 相同都是只有一个值的数据类型。<br><code>null</code> 代表一个 “空对象指针”，这也是为什么使用 <code>typeof null</code> 返回的却是一个 <code>object</code>类型的原因。<br><code>null</code> 对于哪些需要保存为对象但还未真正保存具体对象引用的变量非常有用。也就是说如果定义的变量专门是用来保存对象，那么最好在定义变量的同时初始化为 <code>null</code> 而不是其它值，这样带来的好处体现于，只要检测 <code>null</code> 值就可以得知变量是否保存了一个真正对象的引用。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> obj = <span class="literal">null</span>;</span><br><span class="line"></span><br><span class="line"><span class="keyword">if</span>(<span class="keyword">typeof</span> obj === <span class="string">"object"</span> &amp;&amp; obj != <span class="literal">null</span>)&#123;</span><br><span class="line">    <span class="comment">//....</span></span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>由于 <code>nundefined</code>的值是派生与 <code>null</code>（其实很多的数据类型都是派生与null，因为对象原型链的源头就是null），因此ECMAScript的语法规定在做相等性检测的时候返回固定值<code>true</code>。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="literal">null</span> == <span class="literal">undefined</span> <span class="comment">// true</span></span><br></pre></td></tr></table></figure>
<p>总的来说，null是专门用来初始化需要保存对象引用的变量，这也是 <code>null</code> 与 <code>undefined</code>之间的重要区别。当变量用于保存基本数据类型时，无需要初始化为 <code>undefined</code>，因为默认就是 <code>undefined</code>，而对于用于保存对象的变量 <code>null</code> 则需要显示的初始化，这既是用于区分一个变量是否引用了一个有效的对象，也有助于进一步区分 <code>undefined</code> 与 <code>null</code> 。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> age;</span><br><span class="line"><span class="keyword">var</span> name;</span><br><span class="line"><span class="keyword">var</span> sex;</span><br><span class="line"><span class="keyword">var</span> othre = <span class="literal">null</span>;</span><br><span class="line"></span><br><span class="line">age = <span class="number">26</span>;</span><br><span class="line">name = <span class="string">'gtshen'</span>;</span><br><span class="line">sex = <span class="string">'man'</span>;</span><br><span class="line"></span><br><span class="line">other = &#123;</span><br><span class="line">    height:<span class="number">1.7</span></span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<h3 id="Boolean"><a href="#Boolean" class="headerlink" title="Boolean"></a>Boolean</h3><p><code>boolean</code> 类型的值有两个：true(真)、false(假)，虽然只有两个，但是 ECMAScript中所有类型的值与这个两个值都有等价关系，而想将其它类型的值转换为对相应的Boolean值，其实非常简单，只需要调用 <code>Boolean()</code> 方法即可。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">Boolean</span>(<span class="string">'string'</span>);  <span class="comment">// true</span></span><br><span class="line"><span class="built_in">Boolean</span>(<span class="number">0</span>)          <span class="comment">// false</span></span><br></pre></td></tr></table></figure>
<p>其转换依据如下：</p>
<table>
<thead>
<tr>
<th style="text-align:center">数据类型</th>
<th style="text-align:center">转换为 true</th>
<th style="text-align:center">转换为 false</th>
</tr>
</thead>
<tbody>
<tr>
<td style="text-align:center">String</td>
<td style="text-align:center">不为空</td>
<td style="text-align:center">为空(‘’)</td>
</tr>
<tr>
<td style="text-align:center">Number</td>
<td style="text-align:center">不为0</td>
<td style="text-align:center">0 或者 NAN</td>
</tr>
<tr>
<td style="text-align:center">Object</td>
<td style="text-align:center">任意对象</td>
<td style="text-align:center">null</td>
</tr>
<tr>
<td style="text-align:center">Undefined</td>
<td style="text-align:center">/</td>
<td style="text-align:center">undefined</td>
</tr>
</tbody>
</table>
<p>掌握 <code>Boolean()</code> 方法的转换依据有助于我们明白JavaScript数据类型的隐式转换。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> str = <span class="string">'string'</span>;</span><br><span class="line"><span class="keyword">if</span>(str) alert(<span class="literal">true</span>);</span><br></pre></td></tr></table></figure>
<h3 id="Number"><a href="#Number" class="headerlink" title="Number"></a>Number</h3><p><strong>简介</strong></p>
<p>与其它语言不同的是，在ECMAScript中 <code>Number</code>类型只有一种并且不区分整数与小数。统一采用 IEEE754 标准，使用64位双精度浮点数来表示与存储。只是在运算的时候，会将可以转换为整数的小数以整数的形式进行计算，从而减少小数位数来更好的节省内存空间。</p>
<blockquote>
<p>该标准不仅规定了浮点数的运算规则与表示格式，还规定了一些特殊值，例如正零(+0)与负零(-0)、无穷数(Infinity)以及非数值(NaN)等。</p>
</blockquote>
<p><strong>字面量格式</strong></p>
<p>ECMAScript支持多种字面量格式，除了默认可以直接使用的十进制数，整数还可以通过八进制数以及十六进制的字面量来表示。</p>
<ul>
<li>八进制：(以8为基数，数码范围在：0 ~ 7) 前缀以 <code>0</code> 开头，如果超出数码的范围，则默认转换为十进制数来保存。</li>
<li>十六进制：(以16基数，数码范围在：[0 ~ 9A ~ F] 不区分大小写)，前缀以 <code>0x</code> 开头。</li>
</ul>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">070</span>  <span class="comment">// 八进制   56(D)</span></span><br><span class="line"><span class="number">080</span>  <span class="comment">// 十进制   80(D)</span></span><br><span class="line"><span class="number">0x1A</span> <span class="comment">// 十六进制 26(D)</span></span><br></pre></td></tr></table></figure>
<blockquote>
<p>需要注意的是ECMAScript5中的严格模式并不支持八进制数。</p>
</blockquote>
<p>虽然 ECMAScript 可以支持多种进制数，但是在运算的时候还会自动转换为对应的十进制数。<br>除了支持多种格式的数值字面量，ECMAScript还支持数值的一种表示方法 -&gt; “科学计数法”。</p>
<blockquote>
<p>M×R^E</p>
</blockquote>
<p><code>M</code> 是有效数字，<code>R</code> 是基数，默认为10，<code>E</code> 就是指数。<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">3125000</span> =&gt; <span class="number">3.125</span>×<span class="number">10</span>^<span class="number">6</span> =&gt; <span class="number">3.125E6</span></span><br><span class="line"><span class="number">0.003</span> =&gt; <span class="number">3</span>×<span class="number">10</span>^<span class="number">-3</span> =&gt; <span class="number">3E-3</span></span><br></pre></td></tr></table></figure></p>
<p><strong>NaN</strong><br><code>NaN</code> 即非数值(Not a Number)，用于表示返回一个非正确的数值结果。<br><code>NaN</code> 特点的就是与任何类型的值进行运算，返回的结果仍然是 <code>NaN</code>。任何值与<code>NaN</code> 进行等价判断，返回的结果总会是 <code>false</code>。<br>ECMAScript 提供了一个 <code>isNaN</code> 的方法，可以用来检测当前的值是否是一个 <code>NaN</code> 。是的话，返回<code>true</code>，不是则返回 <code>false</code>。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">isNaN</span>(<span class="literal">NaN</span>)   <span class="comment">// true</span></span><br><span class="line"><span class="built_in">isNaN</span>(<span class="string">'123'</span>) <span class="comment">// false</span></span><br><span class="line"><span class="built_in">isNaN</span>(<span class="literal">true</span>)  <span class="comment">// false</span></span><br><span class="line"><span class="built_in">isNaN</span>(<span class="string">'blue'</span>) <span class="comment">// true</span></span><br></pre></td></tr></table></figure>
<blockquote>
<p><code>isNaN()</code> 方法在具体的检测中也会根据需要进行值的类型转换。</p>
</blockquote>
<p><strong>精度与范围</strong></p>
<p>由于内存容量的限制，以及按照IEEE754规定使用64位二进制位来保存浮点数，因此在 ECMAScript中浮点数的表示范围与精度都是有限的，不可能存的下这个世界上所有的值。</p>
<p>先观察下面这个计算：<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">0.1</span> + <span class="number">0.2</span> <span class="comment">// 0.30000000000000004</span></span><br></pre></td></tr></table></figure></p>
<p>运算的结果超出我们的预料，原本以为结果是 <code>0.3</code>，实际上的结果却是 <code>0.30....4</code>，我们先抛开结果不谈，就说下通过观察这个计算，而得出的一些结论：</p>
<ul>
<li><code>0.300..4</code> 小数点的位数只有17位，这说明了ECMAScript最多只能保持17位小数的精度，超出部分会被舍去。</li>
<li><code>.300.4</code> 结尾是4，结合第一条超出17位会被舍去的结论，这里肯定不仅发生舍去还发生了进位。</li>
<li>是否从第一条与第二条结论中，推导出 ECMAScript底层进行数值运算的时候 <code>0.1</code> 与 <code>0.2</code> 在转换的时候发生了无限不循环的情况呢？</li>
</ul>
<p>经过查证，我们可以得知在很多编程语言（包括ECMAScript）其底层的运算都是基于二进制补码的形式来计算的，所以我们将  <code>0.1</code> 与 <code>0.2</code> 转换为对应的补码：</p>
<figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line">0.1[补] = 0.0001 1001 1001 1001 1001 1001 1001 1001 …（1001无限循环）　</span><br><span class="line">0.2[补] = 0.0011 0011 0011 0011 0011 0011 0011 0011 …（0011无限循环）</span><br></pre></td></tr></table></figure>
<p>接着我们再大致了解下 IEEE754的存储格式，IEEE754会使用双精度(64bit)来存储浮点数，其中符号位占据1位，指数位占据11位，而尾数部分则占据52位，具体如下图所示：</p>
<p><img src="/2018/06/01/professional-javascript-chapter-3/ieee754.png" alt="&#39;js 浮点型&#39;"></p>
<p>这也就直接说明了 <code>0.1</code> 与 <code>0.2</code> 在底层转换为补码的时候，就会因为无限循环而最多只能保存53位尾数的情况下发生了舍去进位，从而丢失精度。而且 <code>0.1</code> 与 <code>0.2</code> 在进行求和的时候，其结果还会再进一步舍去进位，丢失精度，最终在JavaScript中以最多17位小数的形式输出丢失精度的结果。</p>
<p>另外，需要注意的是，IEEE754还规定了，有效数字第一位默认总是1。因此表示精度的尾数前面，还存在着一个隐藏位，固定为1，它不包含在64位浮点数长度之中，也就是说尾数的格式总是：<code>1.xx...xx</code> 的格式，其中 <code>xx...xxx</code> 保存在64位长度之中，最长为52位，所以 JavaScript提供的有效数字(尾数)实际最长为53个二进制位。</p>
<p>这是为什么 <code>Number.MAX_SAFE_INTEGER</code> 的值是 <code>9007199254740991</code> 而 <code>Number.MIN_SAFE_INTEGER</code> 是 <code>-9007199254740991</code> 这是因为受限与尾数的位数，其数值可表示的安全范围则是: [-2^53-1,2^53-1]。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">Number</span>.MAX_SAFE_INTEGER === <span class="built_in">Math</span>.pow(<span class="number">2</span>,<span class="number">53</span>)<span class="number">-1</span>  <span class="comment">//true</span></span><br><span class="line"><span class="built_in">Number</span>.MIN_SAFE_INTEGER === -<span class="built_in">Math</span>.pow(<span class="number">2</span>,<span class="number">53</span>)+<span class="number">1</span> <span class="comment">//true</span></span><br></pre></td></tr></table></figure>
<p>但是实际上，由于尾数前面的隐藏位总为1，所以只要其范围在 <code>(2^52)-1 &lt; x &lt;= (2^53)-1</code> 之内 都可能会出现精度不准的情况。也就是说JavaScript中数值最安全的范围是在 <code>[-(2^52)-1,(2^52)-1]</code> 范围内。</p>
<p>接着再说下JavaScript的最大值与最小值，如果说数值的精度是由尾数的数量来决定的话，那么数值的大小或者说范围就是由指数来决定，IEEE754中，指数的长度是11位，因此指数部分能表示的最大指数就是 <code>2^11-1</code>  即 2047，然后根据规定再取中间值进行偏移，用来表示负指数，也就是说指数的范围是 <code>[-1023,1024]</code>之间，因此，这种存储结构能够表示的数值范围为 <code>2^1024</code> 到 <code>2^-1023</code> ，超出这个范围的数JavaScript便无法表示，只能返回 <code>Infinity</code>，Infinity 根据值的正负情况具有 <code>+Infinity</code> 与 <code>-Infinity</code> 。<br><code>2^1024</code> 转换为科学计数法如下所示：<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">2^1024  = 1.7976931348623157 × 10308 === Number.MAX_VALUE === 1.7976931348623157e+308</span><br></pre></td></tr></table></figure></p>
<p>也因此最小值为 <code>Number.MIN_VALUE = 5e-324</code> 既无限接近于 0。<br>总的来说，对于JavaScript中Number类型有效值的范围以及最大最小值可以通过Number对象的以下属性获得：</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">Number</span>.MIN_VALUE <span class="comment">//最小值</span></span><br><span class="line"><span class="built_in">Number</span>.MAX_VALUE <span class="comment">//最大值</span></span><br><span class="line"><span class="built_in">Number</span>.GATIVE_INFINITY <span class="comment">//负极限</span></span><br><span class="line"><span class="built_in">Number</span>.POSITIVE_INFINITY <span class="comment">//正极限</span></span><br></pre></td></tr></table></figure>
<p>也可以通过以下的方法来判断当前的值是否超出ECMAScript的可表示范围：</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">Number</span>.isFinite(num)</span><br></pre></td></tr></table></figure>
<p>最后，受限于JavaScript对数值的精度与范围的问题，要尽量避免使用JS来运算和处理大数值数据，如果必须要处理的话，也请使用一些第三方的数学库，来规避这些问题，这里推荐的数学库有：Math.js、decimal.js、big.js。</p>
<p><strong>数值转换</strong><br>在 ECMAScript中有三个方法可以显示的将其它类型的值转换为对应的数值型，这三个方法分别是：<code>Number()</code>、<code>parseInt()</code>、<code>parseFloat()</code>。<br><code>Number()</code>方法是对 <code>Number</code> 对象的静态调用，它支持将多种数据格式转换为对应的数值类型，其转换的主要规则（凭据）如下：</p>
<ul>
<li><code>undefined</code> 类型会转换为 <code>NaN</code> 。</li>
<li><code>null</code> 类型会转换为 <code>0</code>，空字符串也会转换为 <code>0</code>。</li>
<li>对于字符串，可以将字符串形式的数值转换为对应的数值数据。（字符串形式的数值指的是只含有数值(0-9)，正负号(+、-)，小数点等字符的字符串），如果字符串中还含有其它特殊字符，则使用 <code>Number()</code> 方法会返回 <code>NaN</code>。需要注意的是如果字符串的开头与结尾含有空格，那么 <code>Number()</code> 会忽略前导与后导空格，但并不会忽略其它部位的空格。</li>
<li>对于布尔型，则会将 <code>true</code> 转换为1 <code>false</code> 转换为 0。</li>
<li>对于数值可以直接转换，对于十六进制或者是八进制的数值，则会转换为对应的十进制(需要注意的是严格模式下不支持八进制)。</li>
<li>对于对象，则调用对象的 <code>valueOf()</code> 方法返回对象的值，如果返回的值仍然是一个引用类型而不是基本类型的值，那么再次调用对象的 <code>toString()</code> 方法，最后再转换。</li>
</ul>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">Number</span>(<span class="string">' 123 '</span>)   <span class="comment">// 123</span></span><br><span class="line"><span class="built_in">Number</span>(<span class="number">0</span>xla);     <span class="comment">// 26</span></span><br><span class="line"><span class="built_in">Number</span>(<span class="string">''</span>)        <span class="comment">// 0</span></span><br><span class="line"><span class="built_in">Number</span>(<span class="string">'123abc'</span>)  <span class="comment">// NaN</span></span><br></pre></td></tr></table></figure>
<p>相比较 <code>Number()</code>方法，<code>parseInt()</code> 以及 <code>parseFloat()</code> 的转换规则要简明的多，只要是符合字符串形式的数值（正负号、小数点，数值0-9）都可以使用<code>parseInt()</code> 或 <code>parseFloat()</code> 转换为对应的十进制数，只是需要注意的是, <code>parseInt()</code> 方法并不识别小数点。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">parseInt</span>(<span class="string">"123"</span>);        <span class="comment">//123</span></span><br><span class="line"><span class="built_in">parseInt</span>(<span class="string">"+123"</span>)        <span class="comment">//123</span></span><br><span class="line"><span class="built_in">parseFloat</span>(<span class="string">'-123.123'</span>); <span class="comment">//-123.123</span></span><br></pre></td></tr></table></figure>
<p>当 <code>parseInt()</code> 与 <code>parseFloat()</code> 在转换字符串形式的数值时，如果遇到特殊的字符，则会停止转换，并返回之前符合条件的值，而如果转换的值是其它类型的值，则返回NaN。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">parseInt</span>(<span class="string">'123.123'</span>);    <span class="comment">//123</span></span><br><span class="line"><span class="built_in">parseInt</span>(<span class="string">'1abc123'</span>);    <span class="comment">//1</span></span><br><span class="line"><span class="built_in">parseFloat</span>(<span class="literal">true</span>)        <span class="comment">//NaN</span></span><br></pre></td></tr></table></figure>
<p>相比较<code>parseFloat()</code>， <code>parseInt()</code> 在转换的时候可以支持将八进制、十六进制转换为对应的十进制。<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">parseInt</span>(<span class="number">070</span>)  <span class="comment">//56</span></span><br><span class="line"><span class="built_in">parseInt</span>(<span class="number">0x1a</span>) <span class="comment">//26</span></span><br></pre></td></tr></table></figure></p>
<p><code>parseInt()</code> 还具有第二个参数，用来指定当前进行转换的数的进制，从而在转换的时候无需为被转换的数添加所需的前缀。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">parseInt</span>(<span class="number">70</span>,<span class="number">8</span>) <span class="comment">//56</span></span><br><span class="line"><span class="built_in">parseInt</span>(<span class="string">'1a'</span>,<span class="number">16</span>) <span class="number">26</span></span><br></pre></td></tr></table></figure>
<blockquote>
<p>同样在ES5严格模式下不支持八进制。</p>
</blockquote>
<h3 id="String"><a href="#String" class="headerlink" title="String"></a>String</h3><p><strong>简介</strong><br><code>String</code> 类型的值就是指由零个或多个使用16位unicode字符保存的字符序列，即字符串。</p>
<p><strong>转义字符</strong><br><code>String</code> 类型的值除了非常普遍的 <code>unicode</code>字符序列，它还有一些具有特殊含义的字面量，比如“转义字符”</p>
<table>
<thead>
<tr>
<th style="text-align:center">转义字符</th>
<th style="text-align:center">含义</th>
</tr>
</thead>
<tbody>
<tr>
<td style="text-align:center">\n</td>
<td style="text-align:center">换行</td>
</tr>
<tr>
<td style="text-align:center">\t</td>
<td style="text-align:center">制表符</td>
</tr>
<tr>
<td style="text-align:center">\b</td>
<td style="text-align:center">退格</td>
</tr>
<tr>
<td style="text-align:center">\r</td>
<td style="text-align:center">回车</td>
</tr>
<tr>
<td style="text-align:center">\f</td>
<td style="text-align:center">分页</td>
</tr>
<tr>
<td style="text-align:center">\[*]</td>
<td style="text-align:center">反转义 ，即不转义。例如：\\ 就是输出斜杠,\’ 输出单引号</td>
</tr>
<tr>
<td style="text-align:center">\xnn</td>
<td style="text-align:center">使用十六进制编码来表示一个字符。例如\x41 就是表示 “A”</td>
</tr>
<tr>
<td style="text-align:center">\unnnn</td>
<td style="text-align:center">使用四位unicode编码来表示一个字符。例如 \u03a3 表示希腊字母Σ</td>
</tr>
</tbody>
</table>
<blockquote>
<p>实际上通过仔细观察可以发现，转义字符都会以斜杠(\) 开头。</p>
</blockquote>
<p><strong>字符串转换</strong><br>将其它数据类型的值显示的转换为 “字符串” 可以直接使用每种数据类型都会自带的 <code>toString()</code> 方法。<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> num = <span class="number">123</span>;</span><br><span class="line"><span class="keyword">var</span> boolean = <span class="literal">true</span></span><br><span class="line"><span class="keyword">var</span> obj = &#123;&#125;; </span><br><span class="line"> </span><br><span class="line">num.toString() <span class="comment">// "123";</span></span><br><span class="line">boolean.toString() <span class="comment">// "true";</span></span><br><span class="line">obj.toString() <span class="comment">//"[object Object]";</span></span><br></pre></td></tr></table></figure></p>
<p>需要注意的是如果是将数值转换为字符串，<code>toString()</code> 方法还可以接受一个参数，即转换为对应的进制数的字符串。<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> num = <span class="number">10</span>;</span><br><span class="line">num.toString(<span class="number">2</span>);  <span class="comment">//"1010"</span></span><br><span class="line">num.toString(<span class="number">8</span>);  <span class="comment">//"12"</span></span><br><span class="line">num.toString(<span class="number">16</span>); <span class="comment">//"a"</span></span><br></pre></td></tr></table></figure></p>
<p>如果不带参数，默认情况下就是直接转换为十进制。<br>为什么每种数据类型都会自带 <code>toString()</code> 方法呢？实际上道理很简单，那就是 ECMAScript中万物皆对象，对象中自带了 <code>toString</code> 方法，而别的数据类型则也是继承至对象，所以便与生俱来自带有 <code>toString</code> 方法。<br>很明显的是像 <code>null</code>，<code>undefined</code> 这两个类型明显不是派生于对象，所以它们并不具有 <code>toString</code> 方法。<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> nl = <span class="literal">null</span>; </span><br><span class="line"><span class="keyword">var</span> un = <span class="literal">undefined</span>;</span><br><span class="line"></span><br><span class="line">nl.toString(); <span class="comment">//Cannot read property 'toString' of null</span></span><br><span class="line">un.toString(); <span class="comment">//Cannot read property 'toString' of undefined</span></span><br></pre></td></tr></table></figure></p>
<p>在此种情况下如果想将 <code>null</code> 与 <code>undefined</code> 再通过对象自带的 <code>toString()</code> 方法转换为字符串明显是行不通的了。<br>解决的方法也很简单，那就是与 <code>Number</code> 类型一样，使用 ECMAScript 自带的静态 <code>String()</code> 方法。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">String</span>(<span class="literal">null</span>); <span class="comment">// "null"</span></span><br><span class="line"><span class="built_in">String</span>(<span class="literal">undefined</span>); <span class="comment">// "undefined"</span></span><br></pre></td></tr></table></figure>
<p>所以在进行字符串转换的时候，为了稳妥起见还是尽可能的多使用 <code>String()</code> 方法。这是因为<code>String()</code> 方法遵循以下转换规则：</p>
<ul>
<li>如果被转换的值具有<code>toString()</code> 方法，则调用其 <code>toString()</code> 方法。</li>
<li>如果是 <code>null</code> 则转换为 “null”。</li>
<li>如果值是 <code>undefined</code> 则转换为　“undefined”。</li>
</ul>
<p>但是使用者需要明白的是，并不是所有的引用类型都会继承至<code>object</code>的<code>toString()</code>方法，也会具有适合自己的<code>toString()</code>方法</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br></pre></td><td class="code"><pre><span class="line"><span class="string">'123'</span>.toString() <span class="comment">// '123'</span></span><br><span class="line"></span><br><span class="line">(<span class="number">255</span>).toString(<span class="number">16</span>) <span class="comment">// 'ff'</span></span><br><span class="line"><span class="built_in">Number</span>(<span class="number">255</span>).toString(<span class="number">16</span>) <span class="comment">//'ff'</span></span><br><span class="line"></span><br><span class="line">[].toString() <span class="comment">// ""</span></span><br><span class="line">[<span class="number">1</span>,<span class="number">2</span>,<span class="number">3</span>].toString() <span class="comment">// "1,2,3"</span></span><br><span class="line"></span><br><span class="line">(&#123;&#125;).toString() <span class="comment">//"[object Object]"</span></span><br></pre></td></tr></table></figure>
<p>当然，你也可以统一的使用某个对象的 <code>toString()</code> 方法来保持统一的转换结果。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">Object</span>.prototype.toString.call(&#123;&#125;)      <span class="comment">//"[object Object]"</span></span><br><span class="line"><span class="built_in">Object</span>.prototype.toString.call([])      <span class="comment">//"[object Array]"</span></span><br><span class="line"><span class="built_in">Object</span>.prototype.toString.call(<span class="string">""</span>)      <span class="comment">//"[object String]"</span></span><br><span class="line"><span class="built_in">Object</span>.prototype.toString.call(<span class="number">1</span>)       <span class="comment">//"[object Number]"</span></span><br><span class="line"><span class="built_in">Object</span>.prototype.toString.call(<span class="literal">true</span>)    <span class="comment">//"[object Boolean]"</span></span><br><span class="line"><span class="built_in">Object</span>.prototype.toString.call(<span class="literal">null</span>)    <span class="comment">//"[object Null]"</span></span><br><span class="line"><span class="built_in">Object</span>.prototype.toString.call(<span class="literal">undefined</span>)<span class="comment">//"[object Object]"</span></span><br></pre></td></tr></table></figure>
<h3 id="Object"><a href="#Object" class="headerlink" title="Object"></a>Object</h3><p>对象类型，也可以称之为“枚举”类型或者是“引用”类型，它是数据与功能的集合，其中数据就是属性，而功能就是方法。<br>在ECMAScript中 <code>Object</code> 对象是其它所有对象的基础，因此<code>Object</code> 对象自带的一些属性方法也必然也会被它的实例化对象所继承。下面就是 <code>Object</code> 对象自带的一些常用属性以及方法：</p>
<p><strong>constructor</strong><br>保存着创建当前对象实例的构造函数。<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">new</span> <span class="built_in">Object</span>().constructor; <span class="comment">// "ƒ Object() &#123; [native code] &#125;"</span></span><br></pre></td></tr></table></figure></p>
<p>这说明当前对象的构造函数就是 <code>Object()</code>。</p>
<p><strong>hasOwnProperty</strong><br>该方法接收一个字符串作为参数，这个字符串就是 key，判断这个key是否在该对象的实例中，而非继承至原型。<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> obj = <span class="keyword">new</span> <span class="built_in">Object</span>();</span><br><span class="line"></span><br><span class="line">obj.custom = <span class="number">1</span>;</span><br><span class="line">obj.hasOwnProperty(<span class="string">'custom'</span>); <span class="comment">// true</span></span><br><span class="line">obj.hasOwnProperty(<span class="string">'custom1'</span>); <span class="comment">// false</span></span><br></pre></td></tr></table></figure></p>
<p><strong>isPrototypeOf(obj)</strong><br>用于检查传入的对象是否是当前对象的原型上。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line"><span class="function"><span class="keyword">function</span> <span class="title">A</span>(<span class="params"></span>)</span>&#123;&#125;;</span><br><span class="line"><span class="keyword">var</span> a = <span class="keyword">new</span> A();</span><br><span class="line"><span class="built_in">console</span>.log(A.prototype.isPrototypeOf(a)); <span class="comment">//true</span></span><br></pre></td></tr></table></figure>
<p><strong>propertyIsEnumerable(propertyName)</strong><br>检查当前对象上指定的属性是否可枚举（for…in循环遍历）。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line"><span class="function"><span class="keyword">function</span> <span class="title">A</span>(<span class="params"></span>)</span>&#123;</span><br><span class="line">    <span class="keyword">this</span>.custom = <span class="number">1</span>;</span><br><span class="line">&#125;;</span><br><span class="line"></span><br><span class="line"><span class="keyword">var</span> a = <span class="keyword">new</span> A();</span><br><span class="line">a.propertyIsEnumerable(<span class="string">'custom'</span>); <span class="comment">//true</span></span><br><span class="line">a.propertyIsEnumerable(<span class="string">'custom1'</span>); <span class="comment">//false</span></span><br></pre></td></tr></table></figure>
<p><strong>toString()</strong><br>返回对象的字符串值。</p>
<p><strong>valueOf()</strong><br>返回对象的值，如果返回的值不是基本数据类型（数值，字符串，布尔值，null）则会继续调用 toString()方法返回对象的字符串值。</p>
<h2 id="运算符"><a href="#运算符" class="headerlink" title="运算符"></a>运算符</h2><p>运算符就是运算符号，在ECMAScript中最神奇的就在于每个运算符都可以适用于多种类型的数据，为了避免程序运行错误，当一个运算符用作于一个不太适当的数据类型时，在运算之前，运算符会先隐式的进行数据类型转换。</p>
<h3 id="自增-自减"><a href="#自增-自减" class="headerlink" title="自增/自减"></a>自增/自减</h3><ul>
<li>当应用于一个字符串形式的数值时，先将其转换为数值再进行自增自减操作。</li>
<li>当应用于一个字符串但是非数值形式时，则转换为NaN</li>
<li>当应用于一个 Boolean 值时，false转换为0，true转换为1，然后进行自增/自减操作。</li>
<li>当应用于<code>null</code>或者是<code>undefined</code>时，<code>null</code> 会被转换为0，<code>undefined</code>为 <code>NaN</code>。</li>
<li>当应用于一个对象(object)时，先调用对象的 <code>valueOf()</code> 方法，如果返回的值不是一个基本类型，则继续调用 <code>toString()</code> 方法，最后进行转换，再自增自减操作。</li>
</ul>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> str = <span class="string">'a'</span>;</span><br><span class="line"><span class="keyword">var</span> str_num = <span class="string">'1.23'</span>;</span><br><span class="line"><span class="keyword">var</span> obj = &#123;&#125;;</span><br><span class="line"></span><br><span class="line">++str;</span><br><span class="line">--str_num;</span><br><span class="line">obj++</span><br></pre></td></tr></table></figure>
<h3 id="一元加减运算符"><a href="#一元加减运算符" class="headerlink" title="一元加减运算符"></a>一元加减运算符</h3><p>一元加减运算符非常符合我们对数学中的正号(+)与负号(-)的认识，只是在 ECMAScript中一元加减运算符除了能改变数值的正负以外，还会对不符合的数据类型进行数值型转换，其转换规则与调用静态方法 <code>Number()</code> 以及自增/自减运算符一致。</p>
<h3 id="位运算符"><a href="#位运算符" class="headerlink" title="位运算符"></a>位运算符</h3><p><strong>简介</strong></p>
<p>在ECMAScript所有的数值都是以 IEEE754标准采用64位二进制进行存储的，但是“位”运算符并不会对64位二进制值进行操作，而是会将64位转换为32位整数再进行运算，最终再将结果转换回64位保存，由于存储和转换的过程对使用者而言都是透明的，所以使用者只能关注到进行运算的32位整数。</p>
<p>在这个32位整数中，有一位为符号位，剩下的31位才是数值位，符号位决定了这个数在计算机中对应二进制数的存储格式，其中0表示正，1表示负，同一个数，如果符号为正，只需要存储其对应的二进制原码，如果为负，便需要将其转换为补码形式进行存储计算。</p>
<p>例如 <code>+8</code> 原码为：</p>
<p><img src="/2018/06/01/professional-javascript-chapter-3/8-ym.png" alt=""></p>
<p>对于正数由于其原、反、补都是相同的，所以无需关注，而对于负数，则求其反码或者是补码会有些繁琐，主要经过以下步骤：</p>
<ol>
<li>求其绝对值的原码，但注意符号位为1</li>
<li>求原码的反码（按位取反，符号位不变）</li>
<li>反码加1</li>
</ol>
<p>例如 <code>-8</code>,求其补码为：</p>
<figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line">（1）- |8| = -1000 = 1...1000</span><br><span class="line">（2） 取反：1...0001</span><br><span class="line">（3） 加1：1...0010</span><br></pre></td></tr></table></figure>
<p>由于ECMAScript的位运算符只能对32位整数进行运算，所以对于超出 <code>2^31-1</code>的数值(符号位占1位)，在运算的时候都会可能出现失去精度的情况。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line">~~<span class="number">2147483647</span> <span class="comment">// 2147483647</span></span><br><span class="line">~~<span class="number">2147483648</span> <span class="comment">// -2147483648</span></span><br><span class="line">~~<span class="number">2147483649</span> <span class="comment">// -2147483647</span></span><br><span class="line">~~<span class="number">4294967296</span> <span class="comment">// 0</span></span><br></pre></td></tr></table></figure>
<p><strong>非、与、或</strong><br>非：~，二进制位（包括符号位）按位取反：<code>~0...1001 =&gt; 1...0110</code>。eg：<code>~9 =&gt; -10</code>。<br>与：&amp;，二进制位同1位1，不同为0。<br>或：|，二进制位有一个1就为1，只有同为0才会为0。</p>
<blockquote>
<p>由于“位”运算符会将64位浮点数转换为32位整数，因此利用 <code>~~</code> 双按位反运算符便可以进行数值的快速取整，其功能类似于parseInt，只是这里进行的是位运算，速度更快。</p>
</blockquote>
<p><strong>异或</strong><br>运算符：^<br>功能：二进制位不同为1，相同为0.<br>示例：<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br></pre></td><td class="code"><pre><span class="line">   101011100</span><br><span class="line">  ^</span><br><span class="line">   010001100</span><br><span class="line">----------------</span><br><span class="line">   111010000</span><br></pre></td></tr></table></figure></p>
<p><strong>左移位</strong><br>运算符：&lt;&lt;<br>功能：将对应操作数的32位二进制数向左移动指定的位数，移动后，右侧空余的位置用0填充。<br>示例：<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">2</span> &lt;&lt; <span class="number">2</span>  <span class="comment">// 8</span></span><br><span class="line"><span class="number">2</span> &lt;&lt; <span class="number">1</span>  <span class="comment">// 4</span></span><br><span class="line"><span class="number">-2</span> &lt;&lt; <span class="number">5</span> <span class="comment">// -64</span></span><br></pre></td></tr></table></figure></p>
<p>需要注意的是“左移”并不会影响操作数的符号位，它移动的只是操作数的绝对值，但是结果符号的依然与操作数的符号相同。</p>
<figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">-64 =&gt; -|-64| =&gt; -(|-64|&lt;&lt;2) =&gt; -256</span><br></pre></td></tr></table></figure>
<p>如果用一组数学公式去计算左移的值的话，应该是：<code>N*~~((2^s))</code>。其中 <code>N</code> 是要移动的操作数，而 <code>s</code> 则是要左移的位数。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">-64</span> &lt;&lt; <span class="number">2</span> == <span class="number">-64</span> * ~~(<span class="built_in">Math</span>.pow(<span class="number">2</span>,<span class="number">2</span>)); <span class="comment">//true</span></span><br></pre></td></tr></table></figure>
<p><strong>有符号右移</strong><br>运算符：&gt;&gt;<br>功能：将对应操作数的32位二进制数向右移动指定的位数，移动后左侧空余的位置用操作数的符号位填充。<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">64</span> &gt;&gt; <span class="number">2</span> <span class="comment">// 16</span></span><br><span class="line"><span class="number">-64</span> &gt;&gt; <span class="number">2</span> <span class="comment">//-16</span></span><br></pre></td></tr></table></figure></p>
<p><strong>无符号右移位</strong><br>运算符：&gt;&gt;&gt;<br>功能：将对应操作数的32位二进制数向右移动指定的位数，移动后左侧空余的位置用0填充。<br>示例：</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">64</span> &gt;&gt;&gt; <span class="number">2</span> <span class="comment">//16</span></span><br></pre></td></tr></table></figure>
<p>如果操作数是正数，那么无符号右移动，与有符号右移完全相同，但是如果操作数是负数，那么结果就会不同了</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">-64</span> &gt;&gt;&gt; <span class="number">5</span> <span class="comment">//67108863</span></span><br></pre></td></tr></table></figure>
<p>出现这个问题的原因，就是因为无符号右移运算符会把负数的二进制数再按照指定的位数右移后，却会以正数的二进制数形式转换为对应的十进制。从而导致结果非常大。</p>
<p>例如 <code>-64</code>的补码是：<code>1111 1111 1111 1111 1111 1111 1100 0000</code> 右移后的结果是 <code>0000 0011 1111 1111 1111 1111 1111 1111</code> 但是在转换为十进制数的时候，并没有减一取反，而是以正数的二进制数直接转换为10进制，所以结果为 <code>67108863</code></p>
<h3 id="逻辑运算符"><a href="#逻辑运算符" class="headerlink" title="逻辑运算符"></a>逻辑运算符</h3><p><strong>非运算符</strong><br>运算符：!<br>功能：“逻辑非”运算符会将操作数取反，最终返回一个布尔值。逻辑非在进行求反之前会调用 <code>Boolean()</code> 这个静态方法将非布尔型的值转换为布尔值，然后进行求反操作。<br>示例：<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line">!<span class="literal">NaN</span> <span class="comment">//true</span></span><br><span class="line">!<span class="literal">undefined</span> <span class="comment">//true</span></span><br><span class="line">!<span class="literal">null</span> <span class="comment">//true</span></span><br></pre></td></tr></table></figure></p>
<p>利用逻辑取反运算符会返回布尔值的这一特性，通过使用双反运算来进行数据类型的转换。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line">!!<span class="literal">NaN</span> <span class="comment">//false</span></span><br><span class="line">!!<span class="literal">undefined</span> <span class="comment">//false</span></span><br><span class="line">!!<span class="literal">null</span> <span class="comment">//false</span></span><br></pre></td></tr></table></figure>
<p><strong>与运算符</strong></p>
<p>运算符：&amp;&amp;<br>功能：“逻辑与”是一个双目运算符，它可以链接多个操作数，并且操作数可以是任何的数据类型，并不一定是非要是布尔值，而且逻辑与运算符的运算结果也并非一定是布尔值，而是满足条件的那个结果。<br>说明：逻辑与存在短路的情况，即只要第一个操作数不为“真(true)”，那么后面的条件就会被忽略。<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="literal">false</span> &amp;&amp; a <span class="comment">// false</span></span><br></pre></td></tr></table></figure></p>
<p>这里变量 <code>a</code> 并未定义，但是再进行逻辑与运算的时候，并不会报错，这是因为在逻辑短路的情况下，后面的代码就不会被执行。<br>对于逻辑与而言，只有当第一个操作数的结果为真的时候，才会向下执行,最终返回符合条件的那个操作数的值，但是一旦遇到值为“假”的操作数，就会中断执行。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line"><span class="literal">true</span> &amp;&amp; <span class="literal">false</span> &amp;&amp; <span class="number">1</span> <span class="comment">// false</span></span><br><span class="line"><span class="literal">NaN</span> &amp;&amp; <span class="string">'true'</span> <span class="comment">//NaN</span></span><br><span class="line"><span class="literal">true</span> &amp;&amp; <span class="number">2</span> <span class="comment">//2</span></span><br></pre></td></tr></table></figure>
<p><strong>或运算符</strong></p>
<p>运算符：||<br>功能：“逻辑或”是一个双目运算符，它可以链接多个操作数，并且操作数可以是任何的数据类型，并不一定非要是布尔值，而且逻辑或运算符的运算结果也并非一定是布尔值，而是满足条件的那个结果。<br>说明：逻辑或存在短路的情况，即只要第一个操作数为“真(true)”，那么后面的条件就会被忽略。<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="literal">true</span> || a <span class="comment">//true</span></span><br></pre></td></tr></table></figure></p>
<p>这里变量 <code>a</code> 并未定义，但是在进行逻或与运算的时候，并不会报错，这是因为再逻辑短路的情况下，后面的代码就不会被执行。<br>对于逻辑或而言，只有当第一个操作数的结果为“假”的时候，才会向下执行,最终返回符合条件的那个操作数的值，但是一旦遇到值为“真”的操作数，就会中断执行。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line"><span class="literal">true</span> || <span class="literal">false</span> <span class="comment">//true</span></span><br><span class="line"><span class="literal">false</span> || <span class="literal">true</span> <span class="comment">//true</span></span><br><span class="line"><span class="literal">false</span> || <span class="literal">false</span> <span class="comment">//false</span></span><br></pre></td></tr></table></figure>
<h3 id="算术运算符"><a href="#算术运算符" class="headerlink" title="算术运算符"></a>算术运算符</h3><p>在算术运算符中，如果操作数的数据类型不一致，也会进行数据类型的隐式转换，只是不同的运算符转换的规则也有一定的不同：</p>
<p><strong>乘/除/求余</strong></p>
<ul>
<li>如果操作数都是数值，则进行常规的<code>乘/除/求余</code>操作，如果所得的结果超过ECMAScript可表示的范围，则返回<code>Infinity</code>。</li>
<li>如果一个操作数的结果是<code>NaN</code>，那么运算结果也将是 <code>NaN</code></li>
<li>如果有一个操作数是非数值类型的，则调用静态 <code>Number()</code> 方法将其转换为数值型，然后进行<code>乘/除/求余</code>操作。</li>
<li>如果零除以零，则结果为 <code>NaN</code></li>
<li>如果一个常规的数值被零除，则结果也为 <code>Infinity</code>。</li>
</ul>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">1</span>*<span class="literal">NaN</span> <span class="comment">//NaN</span></span><br><span class="line"><span class="number">1</span>/<span class="number">0</span>   <span class="comment">//Infinity</span></span><br><span class="line"><span class="number">0</span>/<span class="number">0</span>  <span class="comment">//NaN</span></span><br></pre></td></tr></table></figure>
<p><strong>加减</strong></p>
<ul>
<li>如果操作数都是常规的数值，则进行常规的<code>加</code>操作，如果所得的结果超过ECMAScript可表示的范围，则返回<code>Infinity</code>。</li>
<li>在进行加法运算时，如果有一个操作数是字符串，则将另一个操作也转换为字符串然后进行拼接，如果两个操作数都是字符串则直接进行拼接操作。</li>
<li>在进行加法运算时，如果有一个操作数是数值，另一个操作数是 <code>null</code> 、<code>undefined</code>、布尔值等其中任意一个，则调用<code>Number()</code> 静态方法将它们转换为对应的值后进行运算。</li>
<li>任何数与<code>NaN</code> 进行 <code>加</code> 运算，结果依然为 <code>NaN</code> 。</li>
<li>如果有操作数是一个引用类型，则调用它的 <code>valueOf()</code> 方法获取这个引用类型的基本值，如果值仍然不是一个基本数据类型，则再调用 <code>toString()</code> 方法。</li>
</ul>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">1</span> + <span class="string">'a'</span> <span class="comment">//'1a'</span></span><br><span class="line"><span class="string">'a'</span> + <span class="string">'b'</span> <span class="comment">//'ab'</span></span><br><span class="line"><span class="number">1</span> + <span class="literal">null</span> <span class="comment">//NaN</span></span><br><span class="line"><span class="number">1</span> + <span class="literal">true</span> <span class="comment">//2</span></span><br><span class="line"><span class="number">1</span> + <span class="literal">undefined</span> <span class="comment">//NaN</span></span><br><span class="line"><span class="number">1</span> + <span class="literal">NaN</span> <span class="comment">//NaN</span></span><br><span class="line"><span class="number">1</span> + &#123;&#125; <span class="comment">// "1[object Object]"</span></span><br></pre></td></tr></table></figure>
<p><strong>减法</strong></p>
<ul>
<li>如果操作数都是常规的数值，则进行常规的<code>减</code>操作，如果所得的结果超过 ECMAScript可表示的范围，则返回<code>Infinity</code>。</li>
<li>任何数与<code>NaN</code> 进行 <code>减</code> 运算，结果依然为 <code>NaN</code> 。</li>
<li>如果操作数中有字符、布尔型、<code>null</code>、<code>undefined</code>中的任意一种，在进行减法运算之前，都会先调用 <code>Number()</code> 方法将它们转换为对应的数值，最后再进行减法运算。</li>
<li>如果操作数中有引用类型，则调用它的 <code>valueOf()</code> 方法获取这个引用类型的值，如果值仍然不是一个基本数据类型，则再调用 <code>toString()</code> 方法，接着再使用 <code>Number()</code> 方法将引用对象返回的值再转换为数值，最后再进行减法运算。</li>
</ul>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">1</span> - <span class="literal">null</span> <span class="comment">//NaN</span></span><br><span class="line"><span class="number">1</span> - undefind <span class="comment">//NaN</span></span><br><span class="line"><span class="number">1</span> - <span class="literal">true</span> <span class="comment">//0</span></span><br><span class="line"><span class="number">1</span> - <span class="literal">NaN</span> <span class="comment">//NaN</span></span><br><span class="line"><span class="number">1</span> - &#123;&#125; <span class="comment">//NaN</span></span><br><span class="line"><span class="string">'A'</span> - <span class="string">'b'</span> <span class="comment">//NaN</span></span><br></pre></td></tr></table></figure>
<h3 id="比较运算符"><a href="#比较运算符" class="headerlink" title="比较运算符"></a>比较运算符</h3><p>比较运算符的返回结果是一个布尔型(boolean)。<br>比较运算符有：&gt; 、&gt;= 、&lt; 、&lt;= 、!= 、!==、==、===。<br>比较运算符在计算的时候，也会根据值的不同而不同。<br>在同类型值的情况下：</p>
<ul>
<li>数值之间直接进行大小比较。</li>
<li>字符串之间按字符顺序进行每个字符的 Ascii码进行比较。</li>
<li>布尔值比较，<code>true</code> 大于 <code>false</code></li>
</ul>
<p>如果所比较的值类型并不相同，那么便需要进行一定的隐式转换之后方能进行比较：</p>
<ul>
<li>数值与字符串之间比较，会将字符串转换为数值，如果不是标准的字符串形式数值，则返回 <code>NaN</code>。</li>
<li>当比较的值中含有 <code>NaN</code> 时，除了不相等运算符外任何比较运算符的结果都是 <code>NaN</code>。</li>
<li>当比较的值中含有 <code>null</code>、<code>undefind</code> 时，<code>null</code> 会转换为数值0，而 <code>undefind</code> 转换后为 <code>NaN</code></li>
<li>如果比较的值中有 <code>false</code> 转换为数值0，如果有 <code>true</code> 转换为数值1。</li>
<li>如果比较的值中含有“引用类型”时除了相等运算符外，都会调用该引用对象的 <code>valueOf</code> 方法返回对象的值，如果返回的值依然不是一个基本数据类型，则再进一步调用 <code>toString()</code> 方法，最后再根据值的数据类型是否一致从而分别应用上述规则。如果对象进行相等性检测，则会判断是否具有相同的引用。</li>
</ul>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br></pre></td><td class="code"><pre><span class="line"></span><br><span class="line"><span class="keyword">var</span> obj = &#123;</span><br><span class="line">    valueOf: <span class="function"><span class="keyword">function</span>(<span class="params"></span>) </span>&#123;</span><br><span class="line">        <span class="keyword">return</span> <span class="number">1</span>;</span><br><span class="line">    &#125;</span><br><span class="line">&#125;;</span><br><span class="line"></span><br><span class="line"><span class="built_in">console</span>.log(<span class="number">2</span> &gt; obj);  <span class="comment">//true</span></span><br><span class="line"><span class="built_in">console</span>.log(<span class="literal">null</span> &gt; <span class="number">0</span>)  <span class="comment">//false</span></span><br><span class="line"><span class="built_in">console</span>.log(<span class="literal">null</span> &gt;= <span class="number">0</span>) <span class="comment">//true</span></span><br><span class="line"><span class="built_in">console</span>.log(<span class="literal">null</span> &lt; <span class="number">0</span>)  <span class="comment">//false</span></span><br><span class="line"><span class="built_in">console</span>.log(<span class="literal">null</span> &lt;= <span class="number">0</span>) <span class="comment">//true</span></span><br></pre></td></tr></table></figure>
<p>在比较运算符中，还有一个非常特殊的地方，那就是相等性与不相等性的比较运算符。</p>
<figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br></pre></td><td class="code"><pre><span class="line">== //相等</span><br><span class="line">!= //不相等</span><br><span class="line"></span><br><span class="line">=== //严格相等（全等）</span><br><span class="line">!== //严格不相等（不全等）</span><br></pre></td></tr></table></figure>
<p>“严格相等”、“严格不相等”与普通的“相等”运算符以及“不相等”运算符的最大区别在于前者在进行比较运算的时候，不会发生数据类型的隐式转换。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line"><span class="number">55</span> == <span class="string">'55'</span> <span class="comment">//true</span></span><br><span class="line"><span class="number">55</span> === <span class="string">'55'</span> <span class="comment">//false</span></span><br></pre></td></tr></table></figure>
<p>对于“严格相等”、“严格不相等”运算符实际上并没有什么好说的，无非就是数据类型不会发生转换（使得使用时对结果的预测更加的准确），因此这里主要着重说明下普通的相等运算符与不相等运算符：</p>
<ul>
<li>按照ECMAScript规定，在进行相等运算时 <code>null == undefined</code> 比较结果为 <code>true</code>（注意在严格相等比较中，返回 <code>false</code> 因为它们并不是相同的数据类型）。</li>
<li><code>NaN</code>不相等 <code>NaN</code> 时，结果为 <code>true</code>。<code>NaN != NaN</code>。</li>
<li><code>null</code> 在进行相等运算时也不会进行类型转换：<code>null == 0 //false</code>，这个需要注意下。</li>
<li>如果操作数中含有 “引用类型”时，进行相等性判断，则判断是否为相同的应用，如果是则返回 <code>true</code>或者返回 <code>false</code>。</li>
</ul>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br></pre></td><td class="code"><pre><span class="line"><span class="literal">null</span> == <span class="literal">undefined</span> <span class="comment">//true</span></span><br><span class="line"><span class="literal">null</span> === <span class="literal">undefined</span> <span class="comment">//false</span></span><br><span class="line"></span><br><span class="line"><span class="literal">NaN</span> != <span class="literal">NaN</span> <span class="comment">//true</span></span><br><span class="line"><span class="literal">null</span> == <span class="number">0</span> <span class="comment">//false</span></span><br><span class="line"></span><br><span class="line"><span class="keyword">var</span> a = &#123;&#125;;</span><br><span class="line"><span class="keyword">var</span> b = a;</span><br><span class="line"><span class="keyword">var</span> c = [];</span><br><span class="line"></span><br><span class="line">a == b <span class="comment">//true</span></span><br><span class="line">c == b <span class="comment">//false</span></span><br></pre></td></tr></table></figure>
<p>对于具体的开发，在进行相等性比较，强烈建议使用“严格相等”与“严格不相等”运算符，避免因数据类型的隐式转换而产生错误。</p>
<h3 id="条件运算符"><a href="#条件运算符" class="headerlink" title="条件运算符"></a>条件运算符</h3><p>条件运算符是 ECMAScript中唯一的一个三目运算符</p>
<blockquote>
<p>conditaion ? exp1 : exp2</p>
</blockquote>
<h3 id="赋值运算符"><a href="#赋值运算符" class="headerlink" title="赋值运算符"></a>赋值运算符</h3><p>需要注意的是一些不常见的复合赋值运算符</p>
<blockquote>
<p>&lt;&lt;= 、&gt;&gt;= 、&gt;&gt;&gt;=、%=</p>
</blockquote>
<h3 id="逗号运算符"><a href="#逗号运算符" class="headerlink" title="逗号运算符"></a>逗号运算符</h3><p>逗号运算符主要有两个作用：</p>
<ul>
<li><p>表达式的分隔，例如同时声明多个变量</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> a,b,c;</span><br></pre></td></tr></table></figure>
</li>
<li><p>返回表达式的最后一项值</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> num = (<span class="number">1</span>,<span class="number">2</span>,<span class="number">3</span>,<span class="number">4</span>); <span class="comment">// 4;</span></span><br></pre></td></tr></table></figure>
</li>
</ul>
<h2 id="表达式与语句"><a href="#表达式与语句" class="headerlink" title="表达式与语句"></a>表达式与语句</h2><p>“表达式”通常指的是用运算符号将操作数链接起来可能会产生运算结果的式子。而“语句”则以功能或者行为为基本单位，用来完成某个给定的任务或者是操作，并常以分号为结尾。平常也称“语句”为指令或指令的集合。</p>
<h3 id="表达式语句"><a href="#表达式语句" class="headerlink" title="表达式语句"></a>表达式语句</h3><p>知道“语句”的定义与概念，所以表达式加上分号 <code>;</code> 也可以称之为表达式语句，再详细的划分有：</p>
<ul>
<li>算术表达式语句</li>
<li>移位表达式语句</li>
<li>逻辑表达式语句<br>…</li>
</ul>
<p>除了众多的表达式语句，在ECMAScript中使用关键字来定义的语句还有“条件语句”、“循环语句”、“函数语句”等。</p>
<h3 id="循环语句"><a href="#循环语句" class="headerlink" title="循环语句"></a>循环语句</h3><p>在ECMAScript中组成循环语句的关键字是有：<code>while</code>、<code>for</code>、<code>in</code>、<code>do</code> 等。<br>这些关键字构成的循环语句主要有以下类别：</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">while</span>()&#123;..&#125; </span><br><span class="line"></span><br><span class="line"><span class="keyword">do</span>&#123;&#125;<span class="keyword">while</span>()</span><br><span class="line"></span><br><span class="line"><span class="keyword">for</span>()&#123;..&#125;</span><br><span class="line"></span><br><span class="line"><span class="keyword">for</span>..in</span><br></pre></td></tr></table></figure>
<p>其中 <code>for</code> 循环是最常用到的，其详细格式如下：<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">for</span>(init;condition;loop_express) statement</span><br></pre></td></tr></table></figure></p>
<p>其中的 <code>init</code> 表示初始化一个变量，这个变量也可以在<code>for</code>循环外定义，但是不论是外还是内，因为 ECMAScript5之前都不存在块级作用域，所以循环体的上下文都可以访问到这个变量。<br><code>condition</code> 是循环的条件，只有为 <code>true</code> 时才会进入循环体，而 <code>loop_express</code> 则是用于控制循环条件状态的变更。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">for</span>(<span class="keyword">var</span> i=<span class="number">0</span>;i&lt;<span class="number">10</span>;i++)&#123;</span><br><span class="line">    <span class="built_in">console</span>.log(i++)</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>在 <code>for</code> 循环中，<code>init</code> 与 <code>loop_express</code> 可以留空，在别的地方定义</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> i = <span class="number">0</span>;</span><br><span class="line"><span class="keyword">for</span>(;i&lt;<span class="number">10</span>;)&#123;</span><br><span class="line">    <span class="built_in">console</span>.log(i++)</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>当然，你也可以三个参数都不要，这样实际上就成为了一个死循环</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">for</span>(;;)&#123;&#125;</span><br></pre></td></tr></table></figure>
<p>但是还是强烈建议在定义 <code>for</code> 循环的时候参数都尽量都写在循环之中。<br>在具体的使用上，<code>init</code> 不仅可以用于初始化变量，还可以是一些列用逗号 <code>,</code> 隔开来同时执行的表达式。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">for</span>(<span class="keyword">var</span> i=<span class="number">0</span>,j=<span class="number">10</span>,m=<span class="number">20</span>;i&lt;<span class="number">20</span>;i++)&#123;...&#125;</span><br></pre></td></tr></table></figure>
<p>除了 <code>for</code> 循环外，常用的就数 <code>for..in</code> 循环了，这两者的区别，<code>for</code> 循环可以更好的对数组进行遍历，而 <code>for..in</code> 则可以方便的枚举对象的成员。</p>
<h3 id="标签语句"><a href="#标签语句" class="headerlink" title="标签语句"></a>标签语句</h3><p>格式：<br><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">label:statements</span><br></pre></td></tr></table></figure></p>
<p>需要注意的是 <code>label</code> 并不是关键字，而是代指“标识符”。<br>“标签语句”常用于与 <code>break</code> 或 <code>continue</code> 等关键字结合使用，在代码执行的时候跳转到程序的指定位置。<br>“标签语句”在具体的使用中，常用于多层循环体的内部循环，实现退出整个循环的功能（因为单纯的 <code>break</code> 只能退出当前循环，而无法在多层的循环内部直接中断所有循环）。<br><figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br></pre></td><td class="code"><pre><span class="line">wrap:</span><br><span class="line">    <span class="keyword">for</span>(<span class="keyword">var</span> i=<span class="number">0</span>;i&lt;<span class="number">10</span>;i++)&#123;</span><br><span class="line">        <span class="keyword">for</span>(<span class="keyword">var</span> j=<span class="number">0</span>;j&lt;<span class="number">10</span>;j++)&#123;</span><br><span class="line">            <span class="keyword">if</span>(j==<span class="number">5</span> &amp;&amp; i==<span class="number">5</span>) <span class="keyword">break</span> wrap;</span><br><span class="line">        &#125;</span><br><span class="line">    &#125;</span><br></pre></td></tr></table></figure></p>
<h3 id="with语句"><a href="#with语句" class="headerlink" title="with语句"></a>with语句</h3><p>将指定的对象加入到当前作用域链的源头，从而实现快速的访问对象中的某个成员。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">with</span>(location)&#123;</span><br><span class="line">    <span class="built_in">console</span>.log(href)</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p><code>with</code> 语句虽然可以提供对象成员的便捷访问，但也会导致歧义的产生，如在 <code>with</code> 语句体外定义了一个变量与当前对象的成员重名，便会使自定义的变量无效（当重名时，with语句优先级使用对象的同名方法或属性）。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> href = &#123;&#125;;</span><br><span class="line"><span class="keyword">with</span>(location)&#123;</span><br><span class="line">  <span class="built_in">console</span>.log(href);</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>另外，大量使用 <code>with</code> 语句还会导致性能的下降，并且ES5标准已经规定严格模式不能使用 <code>with</code> 语句。</p>
<h3 id="switch-语句"><a href="#switch-语句" class="headerlink" title="switch 语句"></a>switch 语句</h3><p><code>switch</code> 语句与 <code>if</code> 语句关系很紧密，一般来说，当需要判断的条件在三个以上的时候，推荐使用 <code>swtich</code> 语句，否则建议使用“条件语句”，下面是 <code>switch</code> 语句的具体格式：</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">switch</span>(expression)&#123;</span><br><span class="line">    <span class="keyword">case</span> value :</span><br><span class="line">        statements;</span><br><span class="line">        <span class="keyword">break</span>;</span><br><span class="line">    <span class="keyword">default</span>:</span><br><span class="line">        statements</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>表达式(expression)与每种情况(case)的值(value)可以是任何数据类型的值，不像其它语言中只能是数值，在 ECMAScript中 表达式与 <code>case</code>的值不仅可以是固定的常量，还允许是一个变量：</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> a = <span class="string">'a'</span>;</span><br><span class="line"></span><br><span class="line"><span class="keyword">switch</span>(<span class="string">"a"</span>)&#123;</span><br><span class="line">    <span class="keyword">case</span> <span class="number">1</span>:</span><br><span class="line">        <span class="keyword">break</span>;</span><br><span class="line">    <span class="keyword">case</span> a:</span><br><span class="line">        <span class="keyword">break</span>;</span><br><span class="line">    <span class="keyword">default</span>:</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p><code>case</code> 除了数据类型可以不固定外，还可以是一个表达式，当然匹配的结果就是与表达式的值进行匹配。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">var</span> num = <span class="number">20</span>;</span><br><span class="line"><span class="keyword">switch</span>(<span class="literal">true</span>)&#123;</span><br><span class="line">    <span class="keyword">case</span> num &lt; <span class="number">0</span> : <span class="keyword">break</span>;</span><br><span class="line">    <span class="keyword">case</span> num &gt; <span class="number">5</span> &amp;&amp; num &lt; <span class="number">10</span> : <span class="keyword">break</span>;</span><br><span class="line">    <span class="keyword">case</span> num == <span class="number">10</span> :<span class="keyword">break</span> ;</span><br><span class="line">    <span class="keyword">default</span>:</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<blockquote>
<p>switch语句内部对 <code>expression</code> 与 <code>case</code> 的 <code>value</code> 来进行判断与匹配时，都是以“全等”的方式来进行判断的，所以 switch中值的匹配时不会发生数据类型的转换。</p>
</blockquote>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">switch</span>(<span class="number">10</span>)&#123;</span><br><span class="line">    <span class="keyword">case</span> <span class="string">'10'</span>:<span class="keyword">break</span>;</span><br><span class="line">    defaut:</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>最后还有一个很实用的技巧，如果一个 <code>switch</code> 语句定义在某一个函数的内部，那么可以使用 <code>return</code> 来代替 <code>break</code>，因为 <code>return</code> 在结束函数执行的过程中也就中断了<code>switch</code>语句的执行。</p>
<h3 id="函数语句"><a href="#函数语句" class="headerlink" title="函数语句"></a>函数语句</h3><p>在ECMAScript中如果函数没有明确指定返回值，那么默认返回 <code>undefined</code>。</p>
<p><strong>函数参数</strong></p>
<p>与其它语言一样，ECMAScript中函数的参数也有两种，分别是声明函数时用来接收值的“形参”以及调用函数时传输的“实参”。<br>继承至ECMAScript弱类型的特点，形参与变量的性质实际上是相同的，可以接收任何类型的值，并且形参还不会像变量那样有作用域提升的情况。</p>
<p>相比与其它语言，ECMAScript中函数参数的传输，不是以单个单个参数的形式传输，而是整个以数组的形式进行传输，也就是说无需关注每次传输时参数的具体数量(形参与实参是否一致)。</p>
<p>关于这点，我们可以在函数体中访问 <code>arguments</code> 对象便一目了然</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">arguments</span>.length <span class="comment">// 获取传入参数的数量</span></span><br><span class="line"><span class="built_in">arguments</span>[<span class="number">0</span>];</span><br><span class="line"><span class="built_in">arguments</span>[<span class="number">1</span>];</span><br><span class="line">...</span><br></pre></td></tr></table></figure>
<p><code>arguments</code> 虽然类似与数组，但并不是真的数组，访问其原型链，可以发现继承至 <code>object</code>。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">arguments</span>.__proto__</span><br></pre></td></tr></table></figure>
<p>对于这些类似与数组，而又非数组的值，我们可以很形象的称之为 <code>Like Array</code>。<br>到了这里，我们就可以明白，实际上函数的形参只是帮助我们更好的理解与书写代码，而非必需要有的，在使用 <code>arguments</code> 修改某个参数的值，其对应的形参也会发生改变：</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line"><span class="built_in">arguments</span>[<span class="number">0</span>] = <span class="string">'test'</span>;</span><br><span class="line"><span class="built_in">console</span>.log(a)</span><br></pre></td></tr></table></figure>
<p>但这并不能说明形参就是对 <code>arguments</code> 中某个元素对象的引用，实际上它们俩的内存空间完全是独立的，只是它们的值会具有相互绑定的关系，从而实现相互更新值的特性。我个人认为这是因为 <code>arguments</code> 毕竟是一个对象，在函数体中频繁的访问或修改对象的成员，对于性能的影响也是不可忽略的。</p>
<p>通过 <code>arguments</code> 除了可以访问函数的参数，获取参数的数量，它还有一个非常有用的方法 <code>callee</code>，它与 <code>this</code>类似可以对自身进行引用，因此常用这个方法实现函数的迭代执行。</p>
<figure class="highlight js"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br></pre></td><td class="code"><pre><span class="line"><span class="function"><span class="keyword">function</span> <span class="title">factorial</span>(<span class="params">num</span>)</span>&#123;</span><br><span class="line">    <span class="keyword">if</span> (num &lt;= <span class="number">1</span>)</span><br><span class="line">        <span class="keyword">return</span> <span class="number">1</span>;</span><br><span class="line">    <span class="keyword">else</span></span><br><span class="line">        <span class="keyword">return</span> num * <span class="built_in">arguments</span>.callee(num - <span class="number">1</span>);</span><br><span class="line">&#125;;</span><br></pre></td></tr></table></figure>
<blockquote>
<p>需要注意的是ECMAScript严格模式下 <code>callee</code> 将无法使用。</p>
</blockquote>

      
    </div>
    
    
    

    

    
      <div>
        <div style="padding: 10px 0; margin: 20px auto; width: 90%; text-align: center;">
  <div>坚持原创技术分享，您的支持将鼓励我继续创作！</div>
  <button id="rewardButton" disable="enable" onclick="var qr = document.getElementById('QR'); if (qr.style.display === 'none') {qr.style.display='block';} else {qr.style.display='none'}">
    <span>打赏</span>
  </button>
  <div id="QR" style="display: none;">

    
      <div id="wechat" style="display: inline-block">
        <img id="wechat_qr" src="/images/wx.jpg" alt="Selaginella 微信支付">
        <p>微信支付</p>
      </div>
    

    
      <div id="alipay" style="display: inline-block">
        <img id="alipay_qr" src="/images/alipay.jpg" alt="Selaginella 支付宝">
        <p>支付宝</p>
      </div>
    

    

  </div>
</div>

      </div>
    

    

    <footer class="post-footer">
      
        <div class="post-tags">
          
            <a href="/tags/读书笔记/" rel="tag"># 读书笔记</a>
          
            <a href="/tags/《JavaScript-高级程序设计》/" rel="tag"># 《JavaScript 高级程序设计》</a>
          
        </div>
      

      
      
      

      
        <div class="post-nav">
          <div class="post-nav-next post-nav-item">
            
              <a href="/2018/05/26/professional-javascript-chapter-2/" rel="next" title="《JavaScript 高级程序设计》第二章：在HTML中使用JavaScript">
                <i class="fa fa-chevron-left"></i> 《JavaScript 高级程序设计》第二章：在HTML中使用JavaScript
              </a>
            
          </div>

          <span class="post-nav-divider"></span>

          <div class="post-nav-prev post-nav-item">
            
              <a href="/2018/06/25/nodejs-npm-manager/" rel="prev" title="NodeJS学习：搭建私有NPM">
                NodeJS学习：搭建私有NPM <i class="fa fa-chevron-right"></i>
              </a>
            
          </div>
        </div>
      

      
      
    </footer>
  </div>
  
  
  
  </article>



    <div class="post-spread">
      
    </div>
  </div>


          </div>
          


          

  
    <div class="comments" id="comments">
      <div id="lv-container" data-id="city" data-uid="MTAyMC80MTkxNC8xODQ2MA"></div>
    </div>

  



        </div>
        
          
  
  <div class="sidebar-toggle">
    <div class="sidebar-toggle-line-wrap">
      <span class="sidebar-toggle-line sidebar-toggle-line-first"></span>
      <span class="sidebar-toggle-line sidebar-toggle-line-middle"></span>
      <span class="sidebar-toggle-line sidebar-toggle-line-last"></span>
    </div>
  </div>

  <aside id="sidebar" class="sidebar">
    
    <div class="sidebar-inner">

      

      
        <ul class="sidebar-nav motion-element">
          <li class="sidebar-nav-toc sidebar-nav-active" data-target="post-toc-wrap">
            文章目录
          </li>
          <li class="sidebar-nav-overview" data-target="site-overview-wrap">
            站点概览
          </li>
        </ul>
      

      <section class="site-overview-wrap sidebar-panel">
        <div class="site-overview">
          <div class="site-author motion-element" itemprop="author" itemscope="" itemtype="http://schema.org/Person">
            
              <img class="site-author-image" itemprop="image" src="/uploads/avatar.jpg" alt="Selaginella">
            
              <p class="site-author-name" itemprop="name">Selaginella</p>
              <p class="site-description motion-element" itemprop="description">知识在于积累，努力必有收获</p>
          </div>

          <nav class="site-state motion-element">

            
              <div class="site-state-item site-state-posts">
              
                <a href="/archives/">
              
                  <span class="site-state-item-count">18</span>
                  <span class="site-state-item-name">日志</span>
                </a>
              </div>
            

            
              
              
              <div class="site-state-item site-state-categories">
                
                  <span class="site-state-item-count">5</span>
                  <span class="site-state-item-name">分类</span>
                
              </div>
            

            
              
              
              <div class="site-state-item site-state-tags">
                
                  <span class="site-state-item-count">6</span>
                  <span class="site-state-item-name">标签</span>
                
              </div>
            

          </nav>

          
            <div class="feed-link motion-element">
              <a href="/atom.xml" rel="alternate">
                <i class="fa fa-rss"></i>
                RSS
              </a>
            </div>
          

          
            <div class="links-of-author motion-element">
                
                  <span class="links-of-author-item">
                    <a href="https://github.com/shenguotao2015" target="_blank" title="gitHub">
                      
                        <i class="fa fa-fw fa-github"></i></a>
                  </span>
                
                  <span class="links-of-author-item">
                    <a href="https://gitee.com/Selaginella" target="_blank" title="码云">
                      
                        <i class="fa fa-fw fa-telegram"></i></a>
                  </span>
                
                  <span class="links-of-author-item">
                    <a href="http://www.cnblogs.com/HCJJ/" target="_blank" title="博客园">
                      
                        <i class="fa fa-fw fa-rss-square"></i></a>
                  </span>
                
                  <span class="links-of-author-item">
                    <a href="mailto:sgt_ah@163.com" target="_blank" title="Email">
                      
                        <i class="fa fa-fw fa-envelope"></i></a>
                  </span>
                
            </div>
          

          
          

          
          

          
        </div>
      </section>

      
      <!--noindex-->
        <section class="post-toc-wrap motion-element sidebar-panel sidebar-panel-active">
          <div class="post-toc">

            
              
            

            
              <div class="post-toc-content"><ol class="nav"><li class="nav-item nav-level-2"><a class="nav-link" href="#语法"><span class="nav-number">1.</span> <span class="nav-text">语法</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#标识符"><span class="nav-number">2.</span> <span class="nav-text">标识符</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#严格模式"><span class="nav-number">3.</span> <span class="nav-text">严格模式</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#关键字"><span class="nav-number">4.</span> <span class="nav-text">关键字</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#保留字"><span class="nav-number">5.</span> <span class="nav-text">保留字</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#变量"><span class="nav-number">6.</span> <span class="nav-text">变量</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#数据类型"><span class="nav-number">7.</span> <span class="nav-text">数据类型</span></a><ol class="nav-child"><li class="nav-item nav-level-3"><a class="nav-link" href="#Undefined"><span class="nav-number">7.1.</span> <span class="nav-text">Undefined</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#Null"><span class="nav-number">7.2.</span> <span class="nav-text">Null</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#Boolean"><span class="nav-number">7.3.</span> <span class="nav-text">Boolean</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#Number"><span class="nav-number">7.4.</span> <span class="nav-text">Number</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#String"><span class="nav-number">7.5.</span> <span class="nav-text">String</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#Object"><span class="nav-number">7.6.</span> <span class="nav-text">Object</span></a></li></ol></li><li class="nav-item nav-level-2"><a class="nav-link" href="#运算符"><span class="nav-number">8.</span> <span class="nav-text">运算符</span></a><ol class="nav-child"><li class="nav-item nav-level-3"><a class="nav-link" href="#自增-自减"><span class="nav-number">8.1.</span> <span class="nav-text">自增/自减</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#一元加减运算符"><span class="nav-number">8.2.</span> <span class="nav-text">一元加减运算符</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#位运算符"><span class="nav-number">8.3.</span> <span class="nav-text">位运算符</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#逻辑运算符"><span class="nav-number">8.4.</span> <span class="nav-text">逻辑运算符</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#算术运算符"><span class="nav-number">8.5.</span> <span class="nav-text">算术运算符</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#比较运算符"><span class="nav-number">8.6.</span> <span class="nav-text">比较运算符</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#条件运算符"><span class="nav-number">8.7.</span> <span class="nav-text">条件运算符</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#赋值运算符"><span class="nav-number">8.8.</span> <span class="nav-text">赋值运算符</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#逗号运算符"><span class="nav-number">8.9.</span> <span class="nav-text">逗号运算符</span></a></li></ol></li><li class="nav-item nav-level-2"><a class="nav-link" href="#表达式与语句"><span class="nav-number">9.</span> <span class="nav-text">表达式与语句</span></a><ol class="nav-child"><li class="nav-item nav-level-3"><a class="nav-link" href="#表达式语句"><span class="nav-number">9.1.</span> <span class="nav-text">表达式语句</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#循环语句"><span class="nav-number">9.2.</span> <span class="nav-text">循环语句</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#标签语句"><span class="nav-number">9.3.</span> <span class="nav-text">标签语句</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#with语句"><span class="nav-number">9.4.</span> <span class="nav-text">with语句</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#switch-语句"><span class="nav-number">9.5.</span> <span class="nav-text">switch 语句</span></a></li><li class="nav-item nav-level-3"><a class="nav-link" href="#函数语句"><span class="nav-number">9.6.</span> <span class="nav-text">函数语句</span></a></li></ol></li></ol></div>
            

          </div>
        </section>
      <!--/noindex-->
      

      

    </div>
  </aside>


        
      </div>
    </main>

    <footer id="footer" class="footer">
      <div class="footer-inner">
        <div class="copyright">&copy; 2018 &mdash; <span itemprop="copyrightYear">2019</span>
  <span class="with-love">
    <i class="fa fa-snowflake-o"></i>
  </span>
  <span class="author" itemprop="copyrightHolder">Selaginella<span class="with-love"><i class="fa fa-heart"></i></span>Hexo 强力驱动</span>

  
</div>









        







        
      </div>
    </footer>

    
      <div class="back-to-top">
        <i class="fa fa-arrow-up"></i>
        
      </div>
    

    

  </div>

  

<script type="text/javascript">
  if (Object.prototype.toString.call(window.Promise) !== '[object Function]') {
    window.Promise = null;
  }
</script>




















  
  
    <script type="text/javascript" src="/lib/jquery/index.js?v=2.1.3"></script>
  

  
  
    <script type="text/javascript" src="/lib/fastclick/lib/fastclick.min.js?v=1.0.6"></script>
  

  
  
    <script type="text/javascript" src="/lib/jquery_lazyload/jquery.lazyload.js?v=1.9.7"></script>
  

  
  
    <script type="text/javascript" src="/lib/velocity/velocity.min.js?v=1.2.1"></script>
  

  
  
    <script type="text/javascript" src="/lib/velocity/velocity.ui.min.js?v=1.2.1"></script>
  


  


  <script type="text/javascript" src="/js/src/utils.js?v=5.1.4"></script>

  <script type="text/javascript" src="/js/src/motion.js?v=5.1.4"></script>



  
  


  <script type="text/javascript" src="/js/src/affix.js?v=5.1.4"></script>

  <script type="text/javascript" src="/js/src/schemes/pisces.js?v=5.1.4"></script>



  
  <script type="text/javascript" src="/js/src/scrollspy.js?v=5.1.4"></script>
<script type="text/javascript" src="/js/src/post-details.js?v=5.1.4"></script>



  


  <script type="text/javascript" src="/js/src/bootstrap.js?v=5.1.4"></script>



  


  




	





  





  
    <script type="text/javascript">
      (function(d, s) {
        var j, e = d.getElementsByTagName(s)[0];
        if (typeof LivereTower === 'function') { return; }
        j = d.createElement(s);
        j.src = 'https://cdn-city.livere.com/js/embed.dist.js';
        j.async = true;
        e.parentNode.insertBefore(j, e);
      })(document, 'script');
    </script>
  












  

  <script type="text/javascript">
    // Popup Window;
    var isfetched = false;
    var isXml = true;
    // Search DB path;
    var search_path = "search.xml";
    if (search_path.length === 0) {
      search_path = "search.xml";
    } else if (/json$/i.test(search_path)) {
      isXml = false;
    }
    var path = "/" + search_path;
    // monitor main search box;

    var onPopupClose = function (e) {
      $('.popup').hide();
      $('#local-search-input').val('');
      $('.search-result-list').remove();
      $('#no-result').remove();
      $(".local-search-pop-overlay").remove();
      $('body').css('overflow', '');
    }

    function proceedsearch() {
      $("body")
        .append('<div class="search-popup-overlay local-search-pop-overlay"></div>')
        .css('overflow', 'hidden');
      $('.search-popup-overlay').click(onPopupClose);
      $('.popup').toggle();
      var $localSearchInput = $('#local-search-input');
      $localSearchInput.attr("autocapitalize", "none");
      $localSearchInput.attr("autocorrect", "off");
      $localSearchInput.focus();
    }

    // search function;
    var searchFunc = function(path, search_id, content_id) {
      'use strict';

      // start loading animation
      $("body")
        .append('<div class="search-popup-overlay local-search-pop-overlay">' +
          '<div id="search-loading-icon">' +
          '<i class="fa fa-spinner fa-pulse fa-5x fa-fw"></i>' +
          '</div>' +
          '</div>')
        .css('overflow', 'hidden');
      $("#search-loading-icon").css('margin', '20% auto 0 auto').css('text-align', 'center');

      $.ajax({
        url: path,
        dataType: isXml ? "xml" : "json",
        async: true,
        success: function(res) {
          // get the contents from search data
          isfetched = true;
          $('.popup').detach().appendTo('.header-inner');
          var datas = isXml ? $("entry", res).map(function() {
            return {
              title: $("title", this).text(),
              content: $("content",this).text(),
              url: $("url" , this).text()
            };
          }).get() : res;
          var input = document.getElementById(search_id);
          var resultContent = document.getElementById(content_id);
          var inputEventFunction = function() {
            var searchText = input.value.trim().toLowerCase();
            var keywords = searchText.split(/[\s\-]+/);
            if (keywords.length > 1) {
              keywords.push(searchText);
            }
            var resultItems = [];
            if (searchText.length > 0) {
              // perform local searching
              datas.forEach(function(data) {
                var isMatch = false;
                var hitCount = 0;
                var searchTextCount = 0;
                var title = data.title.trim();
                var titleInLowerCase = title.toLowerCase();
                var content = data.content.trim().replace(/<[^>]+>/g,"");
                var contentInLowerCase = content.toLowerCase();
                var articleUrl = decodeURIComponent(data.url);
                var indexOfTitle = [];
                var indexOfContent = [];
                // only match articles with not empty titles
                if(title != '') {
                  keywords.forEach(function(keyword) {
                    function getIndexByWord(word, text, caseSensitive) {
                      var wordLen = word.length;
                      if (wordLen === 0) {
                        return [];
                      }
                      var startPosition = 0, position = [], index = [];
                      if (!caseSensitive) {
                        text = text.toLowerCase();
                        word = word.toLowerCase();
                      }
                      while ((position = text.indexOf(word, startPosition)) > -1) {
                        index.push({position: position, word: word});
                        startPosition = position + wordLen;
                      }
                      return index;
                    }

                    indexOfTitle = indexOfTitle.concat(getIndexByWord(keyword, titleInLowerCase, false));
                    indexOfContent = indexOfContent.concat(getIndexByWord(keyword, contentInLowerCase, false));
                  });
                  if (indexOfTitle.length > 0 || indexOfContent.length > 0) {
                    isMatch = true;
                    hitCount = indexOfTitle.length + indexOfContent.length;
                  }
                }

                // show search results

                if (isMatch) {
                  // sort index by position of keyword

                  [indexOfTitle, indexOfContent].forEach(function (index) {
                    index.sort(function (itemLeft, itemRight) {
                      if (itemRight.position !== itemLeft.position) {
                        return itemRight.position - itemLeft.position;
                      } else {
                        return itemLeft.word.length - itemRight.word.length;
                      }
                    });
                  });

                  // merge hits into slices

                  function mergeIntoSlice(text, start, end, index) {
                    var item = index[index.length - 1];
                    var position = item.position;
                    var word = item.word;
                    var hits = [];
                    var searchTextCountInSlice = 0;
                    while (position + word.length <= end && index.length != 0) {
                      if (word === searchText) {
                        searchTextCountInSlice++;
                      }
                      hits.push({position: position, length: word.length});
                      var wordEnd = position + word.length;

                      // move to next position of hit

                      index.pop();
                      while (index.length != 0) {
                        item = index[index.length - 1];
                        position = item.position;
                        word = item.word;
                        if (wordEnd > position) {
                          index.pop();
                        } else {
                          break;
                        }
                      }
                    }
                    searchTextCount += searchTextCountInSlice;
                    return {
                      hits: hits,
                      start: start,
                      end: end,
                      searchTextCount: searchTextCountInSlice
                    };
                  }

                  var slicesOfTitle = [];
                  if (indexOfTitle.length != 0) {
                    slicesOfTitle.push(mergeIntoSlice(title, 0, title.length, indexOfTitle));
                  }

                  var slicesOfContent = [];
                  while (indexOfContent.length != 0) {
                    var item = indexOfContent[indexOfContent.length - 1];
                    var position = item.position;
                    var word = item.word;
                    // cut out 100 characters
                    var start = position - 20;
                    var end = position + 80;
                    if(start < 0){
                      start = 0;
                    }
                    if (end < position + word.length) {
                      end = position + word.length;
                    }
                    if(end > content.length){
                      end = content.length;
                    }
                    slicesOfContent.push(mergeIntoSlice(content, start, end, indexOfContent));
                  }

                  // sort slices in content by search text's count and hits' count

                  slicesOfContent.sort(function (sliceLeft, sliceRight) {
                    if (sliceLeft.searchTextCount !== sliceRight.searchTextCount) {
                      return sliceRight.searchTextCount - sliceLeft.searchTextCount;
                    } else if (sliceLeft.hits.length !== sliceRight.hits.length) {
                      return sliceRight.hits.length - sliceLeft.hits.length;
                    } else {
                      return sliceLeft.start - sliceRight.start;
                    }
                  });

                  // select top N slices in content

                  var upperBound = parseInt('1');
                  if (upperBound >= 0) {
                    slicesOfContent = slicesOfContent.slice(0, upperBound);
                  }

                  // highlight title and content

                  function highlightKeyword(text, slice) {
                    var result = '';
                    var prevEnd = slice.start;
                    slice.hits.forEach(function (hit) {
                      result += text.substring(prevEnd, hit.position);
                      var end = hit.position + hit.length;
                      result += '<b class="search-keyword">' + text.substring(hit.position, end) + '</b>';
                      prevEnd = end;
                    });
                    result += text.substring(prevEnd, slice.end);
                    return result;
                  }

                  var resultItem = '';

                  if (slicesOfTitle.length != 0) {
                    resultItem += "<li><a href='" + articleUrl + "' class='search-result-title'>" + highlightKeyword(title, slicesOfTitle[0]) + "</a>";
                  } else {
                    resultItem += "<li><a href='" + articleUrl + "' class='search-result-title'>" + title + "</a>";
                  }

                  slicesOfContent.forEach(function (slice) {
                    resultItem += "<a href='" + articleUrl + "'>" +
                      "<p class=\"search-result\">" + highlightKeyword(content, slice) +
                      "...</p>" + "</a>";
                  });

                  resultItem += "</li>";
                  resultItems.push({
                    item: resultItem,
                    searchTextCount: searchTextCount,
                    hitCount: hitCount,
                    id: resultItems.length
                  });
                }
              })
            };
            if (keywords.length === 1 && keywords[0] === "") {
              resultContent.innerHTML = '<div id="no-result"><i class="fa fa-search fa-5x" /></div>'
            } else if (resultItems.length === 0) {
              resultContent.innerHTML = '<div id="no-result"><i class="fa fa-frown-o fa-5x" /></div>'
            } else {
              resultItems.sort(function (resultLeft, resultRight) {
                if (resultLeft.searchTextCount !== resultRight.searchTextCount) {
                  return resultRight.searchTextCount - resultLeft.searchTextCount;
                } else if (resultLeft.hitCount !== resultRight.hitCount) {
                  return resultRight.hitCount - resultLeft.hitCount;
                } else {
                  return resultRight.id - resultLeft.id;
                }
              });
              var searchResultList = '<ul class=\"search-result-list\">';
              resultItems.forEach(function (result) {
                searchResultList += result.item;
              })
              searchResultList += "</ul>";
              resultContent.innerHTML = searchResultList;
            }
          }

          if ('auto' === 'auto') {
            input.addEventListener('input', inputEventFunction);
          } else {
            $('.search-icon').click(inputEventFunction);
            input.addEventListener('keypress', function (event) {
              if (event.keyCode === 13) {
                inputEventFunction();
              }
            });
          }

          // remove loading animation
          $(".local-search-pop-overlay").remove();
          $('body').css('overflow', '');

          proceedsearch();
        }
      });
    }

    // handle and trigger popup window;
    $('.popup-trigger').click(function(e) {
      e.stopPropagation();
      if (isfetched === false) {
        searchFunc(path, 'local-search-input', 'local-search-result');
      } else {
        proceedsearch();
      };
    });

    $('.popup-btn-close').click(onPopupClose);
    $('.popup').click(function(e){
      e.stopPropagation();
    });
    $(document).on('keyup', function (event) {
      var shouldDismissSearchPopup = event.which === 27 &&
        $('.search-popup').is(':visible');
      if (shouldDismissSearchPopup) {
        onPopupClose();
      }
    });
  </script>





  

  

  

  
  

  

  

  

</body>
</html>
